<template>
  <!-- 瀑布流组件 -->
  <div
    class="relative"
    ref="containerTarget"
    :style="{
      height: containerHeight + 'px'
    }"
  >
    <!-- 渲染数据 -->
    <template v-if="columnWidth && data.length">
      <div
        v-for="(item, index) in data"
        :key="nodeKey ? item[nodeKey] : index"
        class="m-waterfall-item absolute duration-300"
        :style="{
          width: columnWidth + 'px',
          left: item._style?.left + 'px',
          top: item._style?.top + 'px'
        }"
      >
        <slot :item="item" :width="columnWidth" :index="index" />
      </div>
    </template>
    <!-- 加载中 -->
    <div v-else>加载中...</div>
  </div>
</template>

<script setup>
import { computed, nextTick, onMounted, onUnmounted, ref, watch } from "vue"
import { getImgElements, getAllImg, onComplateImgs, getMinHeightColumn, getMinHeight, getMaxHeight } from './utils'

const props = defineProps({
  // 数据源
  data: {
    type: Array,
    required: true
  },
  // 唯一标识 key
  nodeKey: {
    type: String
  },
  // 列数
  column: {
    type: Number,
    default: 2
  },
  // 列间距
  columnSpacing: {
    type: Number,
    default: 20
  },
  // 行间距
  rowSpacing: {
    type: Number,
    default: 20
  },
  // 是否需要进行图片预读取
  picturePreReading: {
    type: Boolean,
    default: true
  }
})

// 容器的总高度 = 最高的这一列的高度
const containerHeight = ref(0)
// 记录每列高度的容器 key：所在列，val：列高
const columnHeightObj = ref({})
/**
 * 构建记录每列高度的对象
 */
const useColumnHeightObj = () => {
  columnHeightObj.value = {}
  for (let i = 0; i < props.column; i++) {
    columnHeightObj.value[i] = 0
  }
}
// 容器实例
const containerTarget = ref(null)
// 容器总宽度（不包含padding、margin、border）
const containerWidth = ref(0)
// 容器的左边距，计算 item 的 left
const containerLeft = ref(0)
/**
 * 计算容器宽度
 */
const useContainerWidth = () => {
  // window.getComputedStyle 是个window对象下的方法，获取当前元素所有最终使用的CSS属性值
  const { paddingLeft, paddingRight } = window.getComputedStyle(containerTarget.value, null)
  // 容器左边距
  containerLeft.value = parseFloat(paddingLeft)
  // 容器宽度
  containerWidth.value = containerTarget.value.offsetWidth - parseFloat(paddingLeft) - parseFloat(paddingRight)
}
// 列宽 = (容器的宽度 - 所有的列间距宽度) / 列数
const columnWidth = ref(0)
// 列间距的合集
const columnSpacingTotal = computed(() => {
  return (props.column - 1) * props.columnSpacing
})
/**
 * 计算列宽
 */
const useColumnWidth = () => {
  // 计算容器宽度
  useContainerWidth()
  // 计算列宽
  columnWidth.value = (containerWidth.value - columnSpacingTotal.value) / props.column
}

onMounted(() => {
  // 计算列宽
  useColumnWidth()
})

// item 高度集合
let itemHeights = []
/**
 * 监听图片加载完成（需要图片预加载）
 */
const waitImgComplate = () => {
  itemHeights = []
  // 拿到所有的元素
  let itemElements = [...document.getElementsByClassName('m-waterfall-item')]
  // 获取到元素的 img 标签
  const imgElements = getImgElements(itemElements)
  // 获取所有 img 标签的图片
  const allImgs = getAllImg(imgElements)
  // 等待图片加载完成
  onComplateImgs(allImgs).then(() => {
    // 图片加载完成
    itemElements.forEach(el => {
      itemHeights.push(el.offsetHeight)
    })
    // 渲染位置
    useItemLocation()
  })
}

/**
 * 不需要图片预加载
 */
const useItemHeight = () => {
  itemHeights = []
  // 拿到所有的元素
  let itemElements = [...document.getElementsByClassName('m-waterfall-item')]
  // 计算 item 高度
  itemElements.forEach(el => {
    itemHeights.push(el.offsetHeight)
  })
  // 渲染位置
  useItemLocation()
}

/**
 * 为每个 item 生产位置属性
 */
const useItemLocation = () => {
  // 遍历数据源
  props.data.forEach((item, index) => {
    // 避免重复计算
    if (item._style) return
    // 生成 style 属性
    item._style = {}
    // left
    item._style.left = getItemLeft()
    // top
    item._style.top = getItemTop()
    // 指定的列高度的自增
    increasinHeight(index)
  })

  // 指定容器的高度
  containerHeight.value = getMaxHeight(columnHeightObj.value)
}

/**
 * 指定列高度自增
 */
const increasinHeight = (index) => {
  // 最小高度所在的列
  const minHeightColumn = getMinHeightColumn(columnHeightObj.value)
  // 使该列自增
  columnHeightObj.value[minHeightColumn] += itemHeights[index] + props.rowSpacing
}

/**
 * 返回下一个 item 的 Left
 */
const getItemLeft = () => {
  // 拿到最小宽度的列表
  const column = getMinHeightColumn(columnHeightObj.value)
  return column * (columnWidth.value + props.columnSpacing) + containerLeft.value
}

/**
 * 返回下一个 item 的 Top
 */
const getItemTop = () => {
  return getMinHeight(columnHeightObj.value)
}

/**
 * 触发计算
 */
watch(() => props.data, (newVal) => {
  nextTick(() => {
    // 第一次获取数据时，构建高度记录容器
    const resetColumnHeight = newVal.every(item => !item._style)
    if (resetColumnHeight) {
      // 构建高度记录容器
      useColumnHeightObj()
    }
    if (props.picturePreReading) {
      waitImgComplate()
    } else {
      useItemHeight()
    }
  })
}, {
  deep: true,
  immediate: true
})

/**
 * 重新构建瀑布流
 */
const reset = () => {
  setTimeout(() => {
    useColumnWidth()
    // 重置所有的定位数据，因为 data 中进行了深度监听，所以该操作会触发 data 的 watch
    props.data.forEach(item => {
      item._style = null
    })
  }, 200)
}

/**
 * 监听列数变化
 */
watch(
  () => props.column,
  () => {
    if (props.picturePreReading) {
      columnWidth.value = 0
      // 数据改变之后，视图改变之后的回调
      reset()
    } else {
      reset()
    }
  }
)

/**
 * 在组件销毁的时候，清除所有的 _style
 */
onUnmounted(() => {
  props.data.forEach(item => {
    delete item._style
  })
})
</script>

<style lang="scss" scoped></style>
